-
Notifications
You must be signed in to change notification settings - Fork 2.3k
feat(gepa): add tool description optimization for multi-agent systems #8928
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
base: main
Are you sure you want to change the base?
feat(gepa): add tool description optimization for multi-agent systems #8928
Conversation
- Add optimize_tool_descriptions parameter (default False) to GEPA - Extract tool descriptions from all nested modules via named_sub_modules() - Apply optimized descriptions in DspyAdapter.build_program() - Enables holistic optimization of tools across main and subagent modules - Tests: 4 new tests, all 16 pass (4 new + 12 existing)
Apologies for accidentally closing #8927 Thank you for the thorough review, @LakshyAAAgrawal! I'll address your feedback:
I'll start working on items 1 and 2 and update the PR soon. Please let me know if you have any specific preferences for the tutorial format! |
Thanks a lot! For the tutorial, I think you can follow the current GEPA tutorial format (load a dataset, show an example from the dataset, build a dspy program, evaluate the baseline program on testset, run GEPA with new optimization settings, show the optimized programs' prompts and tool descriptions, and finally evaluate the optimized program). Hopefully we should be able to see a nice and large gain on agentic tasks with this amazing contribution by you! |
- Add ToolProposer with GenerateImprovedToolDescription signature - Implement routing logic to separate tools from signatures - Tools use ToolProposer, signatures use custom or parent default - Backward compatible: preserves existing custom_instruction_proposer behavior - Add test verifying routing splits components correctly
- Define tool functions outside class for clarity - Match structure of simple ReAct example - Add clear comments explaining architecture - Make code more readable and maintainable
197f077
to
c4f2041
Compare
Hi @LakshyAAAgrawal, I've implemented the tool-specific proposer as requested! Here's what's included: 1. Tool-Specific Proposer Implementation ✅
2. Documentation ✅
Reflection Prompt Design: Before I create a short tutorial (item #3), would you have any feedback on:
Any feedback would be helpful before I invest time in the tutorial. Thank you! |
wait there is a bug in the implementation working on it to fix. Also test has to be fixed. |
…euse Tools now copy ReAct's reflective data with tool-specific annotation instead of complex trajectory extraction. This 15-line approach reuses ReAct's existing context (thoughts, tool calls, observations) and adds focused annotation for each tool. Implementation: - Tools receive full ReAct reflective examples (same trajectory context) - Feedback prefixed: [Optimizing tool: 'X'] for focused optimization - Reflection LM sees complete multi-step execution traces per tool Benefits: - Simpler: 15 lines vs 70+ line extraction approach - Reuses code: No duplicate trajectory formatting logic - Same context: Tools see full ReAct execution traces - Clean: Removed all debug output Tests: - 4 focused tests following GEPA patterns (removed 1 redundant) - 226KB fixture with 34 LM + 6 reflection calls - All tests passing with gpt-5-nano traces Documentation: - Updated GEPA_Advanced.md with implementation details - Explains reflective dataset construction approach
|
||
The `optimize_tool_descriptions` parameter enables GEPA to optimize tool descriptions in addition to signature instructions. This is particularly valuable for ReAct agents and other tool-using systems, where the quality of tool descriptions directly impacts the agent's ability to select appropriate tools for each task. | ||
|
||
Unlike signature instructions that guide reasoning strategies, tool descriptions serve a fundamentally different purpose: they help agents decide **which tool to use** in a given situation. GEPA recognizes this categorical difference and applies a specialized reflection prompt tailored for tool selection decisions. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
which tool to use, when to use it, and how to use it. All three are captured by the description.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Let's avoid the word "fundamentally". One can imagine that all of tool descriptions can (and many times do) simply included in the system prompt itself.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Please also add a corresponding entry in GEPA Overview, that links to this file/section.
|
||
Consider enabling `optimize_tool_descriptions=True` when: | ||
|
||
- **Building ReAct agents**: ReAct agents rely on tool descriptions to make action selection decisions |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
One should consider using this, when they use dspy.Tool anywhere in the DSPy program. Here are a few scenarios for using dspy.Tool:
) | ||
``` | ||
|
||
**Note:** Tool optimization is fully backward compatible. Existing programs without tools, or with `optimize_tool_descriptions=False`, continue to work exactly as before. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I don't think we need to inform users about backward compatibility here. It should be implicit that there should be no behaviour changes for any program not containing dspy.Tool.
dspy/teleprompt/gepa/gepa.py
Outdated
raised if a mismatch in module-level and predictor-level score is detected. | ||
optimize_tool_descriptions: Whether to optimize tool descriptions for modules with tools | ||
(e.g., ReAct agents). When enabled, tool descriptions are included in the optimization | ||
process alongside signature instructions. Default is False. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Add a link to GEPA Advanced/Tool section
dspy/teleprompt/gepa/gepa_utils.py
Outdated
) | ||
|
||
self.propose_new_texts = custom_propose_new_texts | ||
elif self.optimize_tool_descriptions: |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Edge case: What should happen when user tries to provide both a custom proposer, and enables optimize_tool_descriptions
dspy/teleprompt/gepa/gepa_utils.py
Outdated
# Handle signature components - replicate proposer's default behavior | ||
sig_texts = {} | ||
if sig_components: | ||
from gepa.strategies.instruction_proposal import InstructionProposalSignature |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is a slight deviation from this PR, but would be a large enhancement (feel free to ignore):
- Create 2 fields, self.instruction_proposal_signature and self.tool_proposer, which are initialized to the default InstructionProposalSignature and ToolProposerSignature.
- Take an argument from dspy.GEPA that can override the default signature values.
dspy/teleprompt/gepa/gepa_utils.py
Outdated
# Second pass: Process tools by copying ReAct data with annotation | ||
react_module_name = None | ||
for name in ret_d.keys(): | ||
if "react" in name.lower(): |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Is this robust? Might it be better to use isinstance or some other way?
Your task is to write a better description for this tool. | ||
Read the examples carefully and identify patterns in when the tool was used successfully versus when it was misused or overlooked. Identify any domain-specific information about the tool's capabilities or appropriate usage that may not be available to the assistant in the future. The assistant may have developed effective patterns for tool selection - if so, ensure the tool description supports those patterns. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Tool use. Also suggest identifying any failure modes of the tool?
Dear @Ju-usc, This is a great PR. Thanks a lot! I have tried to be overly critical and made too many nits. Feel free to ignore if you disagree with something. Let me know if you'd like me to address anything! Regarding the meta prompt, overall I think it looks great. However, I suggest that as you build the tutorial, you may find that the reflection prompt needs tweaking, or the content exposed in reflective_dataset for the tool may be lacking or need improvement. This is going to be an empirical exercise, which will guide what works in the reflection meta prompts. ! Looking forward to the tutorial on this too! You may already have thoughts about what you'd like to show in the tutorial, but if not, you may consider building off (https://kargarisaac.medium.com/building-and-optimizing-multi-agent-rag-systems-with-dspy-and-gepa-2b88b5838ce2) by @kargarisaac. |
- Add GenerateImprovedToolDescriptionFromFeedback signature documentation - Include tool-aware metric example showing trajectory access - Document tool prefix annotation in feedback - Note component_selector applies to both signatures and tools - Fix 'fundamentally' language per reviewer feedback
- Separate Pass 1 (predictor examples) and Pass 2 (tool aggregation) - Clarify Generated Outputs includes full trajectory for ReAct - Fix feedback annotation format to [Tool 'name' from 'predictor_key'] - Add Component Identification & Proposer Routing section - Explain dual-proposer independence (custom proposer doesn't affect tool proposer) - Use consistent terminology: 'predictor' and 'signature instructions'
Per reviewer feedback, backward compatibility should be implicit
- Add component_selector='all' to optimize all components together - Show how to view optimized tool descriptions - Add example output demonstrating improvement from vague to specific descriptions - Remove unnecessary comments for cleaner examples
- Document why full ReAct trajectory is shared with all tools - Explain rationale: tool interdependencies, selection patterns, workflow context - Add concrete example of optimization benefit - Describe alternative considered (tool-specific filtering) and rejection reasoning - Add future work section on joint tool optimization - Present two architectural approaches: separate proposer vs extending ReAct proposer - Include implementation details, benefits, challenges, and decision rationale
- Add Tool Description Optimization section to GEPA overview.md with link to advanced guide - Add documentation link to optimize_tool_descriptions parameter in gepa.py - Addresses reviewer feedback to make tool optimization more discoverable
…ion details - Restructure 'When to Use' as numbered list (1-5) per reviewer feedback - Move section after implementation details for better flow - Remove tool: prefix implementation detail from component identification - Explain tool discovery via ReAct modules in user-friendly terms - Add custom proposer compatibility clarification - Address optional PR feedback items (11 & 13)
Hi @LakshyAAAgrawal, Thanks for the detailed review! I tried my best to address all the critical points, though I may have missed something or introduced new issues. Feel free to add any thoughts! I also have a few questions for you. Implementation: Documentation: Key design decision: For reflective dataset, we share/deepcopy the complete ReAct trajectory with each tool (not just tool-specific segments) because tools need full context to understand selection patterns and workflow dependencies. This trades token efficiency for richer optimization. Future work - would love your thoughts: For reflective dataset for tools, I documented two approaches to reduce token duplication (see code comments for details): joint optimization with ReAct instruction by a separate proposer, or extending the instruction proposer to include tools. The second seems more elegant. What's your take - should we pursue one of these, or is the current approach good enough? Testing:
Ready for re-review! |
- Add note that proposed architecture details may change - Expand challenges with counterpoints and questions - Mark implementation notes as optional to avoid overengineering
|
||
GEPA discovers tools by traversing ReAct modules and extracting their associated `dspy.Tool` instances. Once identified, GEPA routes components to appropriate proposers: | ||
- **Signature instructions** → Custom instruction proposer (if provided) OR default GEPA proposer | ||
- **Tool descriptions** → Built-in `ToolProposer` (always used, not customizable) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is customizable, right? If not, shouldn't we make this customizable too?
This is great, and the right approach. I would just like to highlight that there may be one issue: In react, the trajectory often ends up containing repeated prefixes, i.e., let's say the actual trajectory is {system prompt, user message 1, assistant 1, tool 1, assistant 2, tool 2, assistant 3, end}, then the react trajectory will contain a list of 7 invocations, each containing prefix of size 1 through 7. We can avoid showing all prefixes, and just show the final trajectory. Could you please ensure we aren't running into that issue here? Relevant gepa-ai/gepa#97 |
and what misuse the feedback exposed. Keep the tool's voice and only change what the | ||
evidence justifies. | ||
Return a refined description that helps the assistant quickly recognize good |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Have you had an opportunity to test this prompt for any workload? I would be glad if you could run this prompt just once with a trajectory on any agentic task, and see the tool description it produces.
Description
Adds tool description optimization capability to GEPA optimizer for multi-agent systems.
When
optimize_tool_descriptions=True
, GEPA now:named_sub_modules()
)This enables holistic optimization where both agent reasoning (signatures) and tool usage (descriptions) are improved based on end-to-end execution traces.
Issue
Closes #8706
Changes
optimize_tool_descriptions
parameter to GEPA (defaultFalse
)named_sub_modules()
traversal incompile()
DspyAdapter.build_program()
Usage Example
Backward Compatibility
✅ Fully backward compatible - default
optimize_tool_descriptions=False
Tests